第二章 正则表达式位置匹配攻略

如何匹配位置呢?

^ $ \b \B (?=p) (?!p)

^$

^(脱字符)匹配开头,在多行匹配中匹配行开头。

$(美元符号)匹配结尾,在多行匹配中匹配行结尾。

比如我们把字符串的开头和结尾用”#”替换(位置可以替换成字符的!):

1
2
3
var result = "hello".replace(/^|$/g, '#');
console.log(result);
// => "#hello#"

多行匹配模式时,二者是行的概念,这个需要我们的注意:

1
2
3
4
5
6
7
var result = "I\nlove\njavascript".replace(/^|$/gm, '#');
console.log(result);
/*
#I#
#love#
#javascript#
*/
\b\B

b是单词边界,具体就是\w\W之间的位置,也包括\w^之间的位置,也包括\w$之间的位置。

比如一个文件名是”[JS] Lesson_01.mp4”中的\b,如下:

1
2
3
var result = "[JS] Lesson_01.mp4".replace(/\b/g, '#');
console.log(result);
// => "[#JS#] #Lesson_01#.#mp4#"

为什么是这样呢?这需要仔细看看。
首先,我们知道,\w是字符组[0-9a-zA-Z_]的简写形式,即\w是字母数字或者下划线的中任何一个字符。而\W是排除字符组[^0-9a-zA-Z_]的简写形式,即\W\w以外的任何一个字符。
此时我们可以看看”[#JS#] #Lesson_01#.#mp4#”中的每一个”#”,是怎么来的。

  • 第一个”#”,两边是”[“与”J”,是\W\w之间的位置。
  • 第二个”#”,两边是”S”与”]”,也就是\w\W之间的位置。
  • 第三个”#”,两边是空格与”L”,也就是\W\w之间的位置。
  • 第四个”#”,两边是”1”与”.”,也就是\w\W之间的位置。
  • 第五个”#”,两边是”.”与”m”,也就是\W\w之间的位置。
  • 第六个”#”,其对应的位置是结尾,但其前面的字符”4”是\w,即\w$之间的位置。

知道了\b的概念后,那么\B也就相对好理解了。
\B就是\b的反面的意思,非单词边界。例如在字符串中所有位置中,扣掉\b,剩下的都是\B的。

具体说来就是\w\w\W\W^\W\W$之间的位置。比如上面的例子,把所有\B替换成”#”:

1
2
3
var result = "[JS] Lesson_01.mp4".replace(/\B/g, '#');
console.log(result);
// => "#[J#S]# L#e#s#s#o#n#_#0#1.m#p#4"
(?=p)(?!p)

(?=p),其中p是一个子模式,即p前面的位置。

比如(?=l),表示’l’字符前面的位置,例如:

1
2
3
var result = "hello".replace(/(?=l)/g, '#');
console.log(result);
// => "he#l#lo"

(?!p)就是(?=p)的反面意思,比如:

1
2
3
var result = "hello".replace(/(?!l)/g, '#');
console.log(result);
// => "#h#ell#o#"

二者的学名分别是positive lookahead和negative lookahead。
中文翻译分别是正向先行断言和负向先行断言。
ES6中,还支持positive lookbehind和negative lookbehind。
具体是(?<=p)和(?<!p)。
也有书上把这四个东西,翻译成环视,即看看右边或看看左边。
但一般书上,没有很好强调这四者是个位置。

位置的特性

把位置理解空字符,是对位置非常有效的理解方式

1
2
3
4
5
6
7
var result = /^^hello$$$/.test("hello");
console.log(result);
// => true

var result = /(?=he)^^he(?=\w)llo$\b\b$/.test("hello");
console.log(result);
// => true

相关案例

1
2
// 不匹配任何东西的正则
var reg = /.^/;
1
2
3
4
// 弄出最后一个逗号
var result = "12345678".replace(/(?=\d{3}$)/g, ',')
console.log(result);
// => "12345,678"
1
2
3
4
// 弄出所有的逗号 因为逗号出现的位置,要求后面3个数字一组,也就是\d{3}至少出现一次 此时可以使用量词+
var result = "12345678".replace(/(?=(\d{3})+$)/g, ',')
console.log(result);
// => "12,345,678"
1
2
3
4
5
6
7
8
9
// 不是期望的结果
var result = "123456789".replace(/(?=(\d{3})+$)/g, ',')
console.log(result);
// => ",123,456,789"

// 匹配开头可以使用^,但要求这个位置不是开头 (?!^)
var result = "123456789".replace(/(?!^)(?=(\d{3})+$)/g, ',')
console.log(result);
// => ",123,456,789"
1
2
3
4
5
6
7
// 如果要把"12345678 123456789"替换成"12,345,678 123,456,789"。
// 此时我们需要修改正则,把里面的开头^和结尾$,替换成\b
var result = "12345678 123456789".replace(/(?!\b)(?=(\d{3})+\b)/g, ',')
console.log(result);

// 其实(?!\b)说的就是\B
var reg = /\B(?=(\d{3})+\b)/g;
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
// 密码长度6-12位,由数字、小写字符和大写字母组成
var reg = /^[0-9A-Za-z]{6,12}$/;

// 要求的必须包含数字 可以用(?=.*[0-9])来做
var reg = /(?=.*[0-9])^[0-9A-Za-z]{6,12}$/;

// 同时包含数字和小写字母,可以用(?=.*[0-9])(?=.*[a-z])来做
var reg = /(?=.*[0-9])(?=.*[a-z])^[0-9A-Za-z]{6,12}$/;

// 必须至少包括2种字符
var reg = /((?=.*[0-9])(?=.*[a-z])|(?=.*[0-9])(?=.*[A-Z])|(?=.*[a-z])(?=.*[A-Z]))^[0-9A-Za-z]{6,12}$/;

// (?=.*[0-9])表示该位置后面的字符匹配.*[0-9],即,有任何多个任意字符,后面再跟个数字

// 另外一种解法: “至少包含两种字符”的意思就是说,不能全部都是数字,也不能全部都是小写字母,也不能全部都是大写字母。
var reg = /(?!^[0-9]{6,12}$)(?!^[a-z]{6,12}$)(?!^[A-Z]{6,12}$)^[0-9A-Za-z]{6,12}$/;

以上信息仅做笔记记录,如有冒犯,请联系我。感谢老姚,附作者掘金地址链接: https://juejin.im/post/59cc61176fb9a00a437b290b